<template>
  <div>
    <!-- 预览文章分类 -->
    <el-card class="box-card">
      <div slot="header" class="clearfix header-box">
        <span>文章分类</span>
        <el-button type="primary" size="mini" @click="addCateShowDailogFn">添加分类</el-button>
      </div>
      <!-- 分类数据表格 -->
      <el-table :data="cateList" style="width: 100%" border stripe>
        <!-- type是table-column内置属性，你告诉他用index,意思就是第一个单元格用索引值 -->
        <el-table-column type="index" label="序号" width="100"></el-table-column>
        <el-table-column prop="cate_name" label="分类名称"></el-table-column>
        <el-table-column prop="cate_alias" label="分类别名"></el-table-column>
        <el-table-column label="操作">
          <!-- scope对象：{ row:行对象} -->
          <template v-slot="scope">
            <el-button type="primary" size="mini" @click="updateCateBtnFn(scope.row)">修改</el-button>
            <el-button type="danger" size="mini" @click="delCateFn(scope.row)">删除</el-button>
          </template>
        </el-table-column>
      </el-table>
    </el-card>

    <!-- 添加文章分类-对话框
      el-dialog  对话框组件
      title：左上角标题
      visible：控制对话框是否显示(右侧vue变量为true显示，为false隐藏)
      visible.sync:组件内检测到对话框关闭(点击蒙层，按esc,右上角x),它会回传false给vue变量
      before-close: 对话框关闭前回调，可以在内部使用done()关闭弹窗，什么都不调用就会阻止对话框关闭

      扩展知识点：.sync是Vue2.3版本新增
      // .sync类似于v-model，可以给子组件实现"双向数据绑定"
      // 复习：v-model如何给子组件实现双向数据绑定(父Vue变量要传给子组件，子组件传出的值也要自动绑定到父Vue变量上)
      // v-model本质：给所在的标签绑定：value="Vue变量" @input="val => Vue变量=val"
      // <标签 v-model="Vue变量"></标签>
      // 运行时如下
      // <标签：value="Vue变量" @input=val => Vue变量 = val></标签>
      // Vue2里面一个标签上v-model只能用一次，Vue3里可以用多次

      // .sync本质：给所在的标签绑定：props属性名="Vue变量" @update:props属性名="val => Vue变量 = val"
      // <标签 :visible.sync="Vue变量"></标签>
      // 运行时如下
      // <标签 :visible="Vue变量" @update:visible="val => Vue变量 = val"></标签>
      // 子组件内子传父的时候 this.$emit('update:visible',值)
      // vue2里面可以用多次，Vue3里面移除了
     -->
    <el-dialog
      title="提示"
      :visible.sync="dialogVisible"
      width="30%"
      @close="dialogCloseFn"
      >
      <!-- 添加的表单 -->
      <el-form
      :model="addForm"
      :rules="addRules"
      ref="addRef"
      label-width="80px"
      >
        <el-form-item label="分类名称" prop="cate_name">
          <el-input v-model="addForm.cate_name" minlength="1" maxlength="10"></el-input>
        </el-form-item>
        <el-form-item label="分类别名" prop="cate_alias">
          <el-input v-model="addForm.cate_alias" minlength="1" maxlength="15"></el-input>
        </el-form-item>
      </el-form>
      <span slot="footer" class="dialog-footer">
        <el-button @click="cancelFn">取 消</el-button>
        <el-button type="primary" @click="confirmFn">确 定</el-button>
      </span>
    </el-dialog>
  </div>
</template>

<script>
// 经验：如果用同一个按钮，想要做状态区分
// 1. 定义一个标记变量isEdit(true编辑，false新增),还要定义本次要编辑的数据唯一id值，editId
// 2. 在点击修改的时候，isEdit改为true,editId保存要修改的数据id
// 3. 在点击新增按钮的时候，isEdit改为false，editId置空
// 4. 在点击保存按钮时(确定按钮时)，就可以用isEdit变量做区分了
import { getArtCateListAPI, saveArtCateAPI, updateArtCateAPI, delArtCateAPI } from '@/api'
export default {
  name: 'ArtCate',
  data () {
    return {
      cateList: [], // 文章分类数组
      dialogVisible: false,
      addForm: { // 添加表单的数据对象
        cate_name: '',
        cate_alias: ''
      },
      addRules: { // 添加表单的验证规则对象
        cate_name: [
          { required: true, message: '请输入分类名称', trigger: 'blur' },
          { pattern: /^\S{1,10}$/, message: '分类名必须是1-10位的非空字符', trigger: 'blur' }
        ],
        cate_alias: [
          { required: true, message: '请输入分类别名', trigger: 'blur' },
          { pattern: /^[a-zA-Z0-9]{1,15}$/, message: '分类别名必须是1-15位的字母数字', trigger: 'blur' }
        ]
      },
      isEdit: false, // true为编辑状态，false新增状态
      editId: '' // 保存正在要编辑的数据id值
    }
  },
  created () {
    // 获取文章分类列表
    this.getArtCateFn()
  },
  methods: {
    // 获取文章分类列表
    async getArtCateFn () {
      const res = await getArtCateListAPI()
      this.cateList = res.data.data
    },
    // 添加分类按钮点击事件 ->为了让对话框出现
    addCateShowDailogFn () {
      this.isEdit = false // 变回新增状态标记
      this.editId = ''
      this.dialogVisible = true
    },
    // 对话框确定按妞->点击事件->让对话框消失/调用保存文章分类接口
    confirmFn () {
      this.$refs.addRef.validate(async valid => {
        if (valid) {
          // 通过校验
          if (this.isEdit) {
            // 要修改
            // this.addForm.id = this.editId // 把要编辑的文章分类id添加到对象上
            // updateArtCateAPI(this.addFrom)
            const { data: res } = await updateArtCateAPI({ id: this.editId, ...this.addForm })
            if (res.code !== 0) return this.$message.error(res.message)
            this.$message.success(res.message)
          } else {
            // 要新增
            const { data: res } = await saveArtCateAPI(this.addForm)
            if (res.code !== 0) return this.$message.error(res.message)
            this.$message.success(res.message)
          }

          // 再重新请求一次文章列表，拿到最新的数据，让表格更新
          // 生命周期的方法比如created,不会挂载到this身上，无法this.created
          this.getArtCateFn()
          this.dialogVisible = false
        } else {
          return false
        }
      })
    },
    // 对话框取消按钮->点击事件
    cancelFn () {
      this.dialogVisible = false
    },
    // 对话框-关闭时的回调
    dialogCloseFn () {
      this.$refs.addRef.resetFields()
    },
    // 修改分类按钮->点击事件(先做数据回显)
    updateCateBtnFn (obj) {
      // obj的值：{ id:文章分类id, cate_name:文章分类名字, cate_alias:文章分类别名 }
      this.isEdit = true
      this.editId = obj.id
      this.dialogVisible = true

      // 让el-dialog第一次挂载el-form时，先用addForm空字符串初始值绑定到内部，后续用作resetFields重置
      // 所以让真实DOM先创建并在内部绑定好“复制 ”好初始值
      // 接着我们真实DOM更新绑定好了，再数据回显
      // 注意：我们给v-model对象赋值只是影响当前显示的值，不会印象到resetFields复制的初始值
      this.$nextTick(() => {
        // 数据回显(回填)
        this.addForm.cate_name = obj.cate_name
        this.addForm.cate_alias = obj.cate_alias
      })
    },
    // 删除分类 -> 点击事件
    async delCateFn (obj) {
      // obj: { id, cate_name, cate_alias }
      const { data: res } = await delArtCateAPI(obj.id)
      if (res.code !== 0) return this.$message.error(res.message)
      this.$message.success(res.message)

      // 删除后，再调用方法去后台再获取最新的数据列表
      this.getArtCateFn()
    }
  }
}

// 小bug:（el-form 和 el-dialog 和 数据回显 同时使用，有bug）
// 第一次打开网页，先点击修改，再点击新增，发现输入框竟然有值
// 原因:点击修改后，关闭对话框的时候，置空失效了
// 具体分析：resetFields() 出现问题
// 线索1：Dialog 的内容是懒渲染，即再第一次被打开之前，传入的默认 solt 不会被渲染到DOM上
// 线索2：vue数据改变(先执行同步所有) 再去更新DOM(异步代码)

// 无问题:第一次打开网页，先点击新增按纽 -> dialog出现 ->
// el-form组件第一次挂载（关联的addForm对象的属性的值为空字符串) el-form组件内绑定了初始值，所以后续调用resetFields的时候，它可以用到空字符串初始值来重置

// 有问题:第一次打开网页，先点击修改按钮 -> 虽然dialog变量为true了但是同步代码把addForm对象里赋值了(默认值)
// -> DOM更新异步- dialog出现 -> el-form组件第一次挂载(使用addForm内置做回显然后第一次e1-form内绑定了初始值(有值) -> 以后做重置，它就用绑定的带值的做重置

// 解决：让数据回显慢点执行

</script>

<style lang="less" scoped>
.header-box {
  display: flex;
  justify-content: space-between;
  align-items: center;
}
</style>
